BZOJ2095【POI2010】Bridges <二分+网络流>

Problem

【POI2010】Bridges


Description

为了减肥,来到了瘦海。
这是一个巨大的海,海中有 个小岛,小岛之间有 座桥连接,两个小岛之间不会有两座桥,并且从一个小岛可以到另外任意一个小岛。
现在 想骑单车从小岛 出发,骑过每一座桥,到达每一个小岛,然后回到小岛
霸中同学为了让 减肥成功,召唤了大风,由于是海上风变得十分大,经过每一座桥都有不可避免的风阻碍 十分 ,于是用泡芙贿赂了你,希望你能帮他找出一条承受的最大风力最小的路线。

Input

第一行为两个用空格隔开的整数
接下来 行,每行为由空格隔开的 个整数 ,第 行表示第 座桥连接小岛 ,从 承受的风力为 ,从 承受的风力为

Output

如果无法完成减肥计划,则输出NIE,否则第一行输出最大风力的最小值。

Sample Input

1
2
3
4
5
4 4
1 2 2 4
2 3 3 4
3 4 4 4
4 1 5 4

Sample Output

1
4

Explanation

HINT

,
对于每座桥,保证 , ,

标签:二分答案 网络流 欧拉回路

Solution

二分答案+网络流判定混合图欧拉回路

二分答案,找出符合条件的边集,原问题变为判断混合图中是否存在欧拉回路。
用经典的网络流判定混合图欧拉回路的建模方式:

  • 给所有无向边随机定向,若 的无向边定向为 ,则连边 流量为
  • 按照定向后的图计算每个点的入度 和出度
  • 对于每个点 ,若 ,则连边 流量为 ;若 ,则连边 流量为

建图后跑一遍 ,若满载,则存在欧拉回路。

大致原理:欧拉回路的充要条件是存在一种定向使得每个点的入度等于出度。随机定向后,连接“ ”的边,这样走这条边可以使得 的出度 的入度 ,达到边反向的效果。而一个点入度每 ,出度均要 ,所以分正负从 或向 连长为 的边,这样满载时这个点一定有

Code

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
#include <bits/stdc++.h>
#define MAX_N 1000
#define MAX_M 10000
#define mid ((l+r)>>1)
#define INF 0x3f3f3f3f
using namespace std;
template <class T> inline void read(T &x) {
x = 0; int c = getchar(), f = 1;
for (; !isdigit(c); c = getchar()) if (c == 45) f = -1;
for (; isdigit(c); c = getchar()) (x *= 10) += f*(c-'0');
}
int n, m, into[MAX_N+5], outo[MAX_N+5];
int s, t, cnt, d[MAX_N+5], pr[MAX_N+5], cr[MAX_N+5];
struct node {int v, c, nxt;} E[MAX_M+5];
struct edge {int u, v, x, y;} e[MAX_M+5];
void init() {s = 0, t = n+1, cnt = 0, memset(pr, -1, sizeof pr);}
void insert(int u, int v, int c) {E[cnt] = (node){v, c, pr[u]}, pr[u] = cnt++;}
void addedge(int u, int v, int c) {insert(u, v, c), insert(v, u, 0);}
bool BFS() {
queue <int> que; que.push(s);
memset(d, -1, sizeof d), d[s] = 0;
while (!que.empty()) {
int u = que.front(); que.pop();
for (int i = pr[u]; ~i; i = E[i].nxt) {
int v = E[i].v, c = E[i].c;
if (~d[v] || !c) continue;
d[v] = d[u]+1, que.push(v);
}
}
return ~d[t];
}
int DFS(int u, int flow) {
if (u == t) return flow; int ret = 0;
for (int &i = pr[u]; ~i; i = E[i].nxt) {
int v = E[i].v, c = E[i].c;
if (d[u]+1 != d[v] || !c) continue;
int tmp = DFS(v, min(flow, c));
E[i].c -= tmp, E[i^1].c += tmp;
flow -= tmp, ret += tmp;
if (!flow) break;
}
if (!ret) d[u] = -1; return ret;
}
void cpy() {for (int i = s; i <= t; i++) cr[i] = pr[i];}
void rec() {for (int i = s; i <= t; i++) pr[i] = cr[i];}
int Dinic() {int ret = 0; cpy(); while (BFS()) ret += DFS(s, INF), rec(); return ret;}
bool chk(int tans) {
int tot = 0; init();
for (int i = 1; i <= m; i++) if (e[i].x > tans) return false;
memset(into, 0, sizeof into), memset(outo, 0, sizeof outo);
for (int i = 1; i <= m; i++) into[e[i].v]++, outo[e[i].u]++;
for (int i = 1; i <= m; i++) if (e[i].y <= tans) addedge(e[i].v, e[i].u, 1);
for (int i = 1; i <= n; i++) if (into[i]%2 != outo[i]%2) return false;
for (int i = 1; i <= n; i++) tot += max(0, (outo[i]-into[i])>>1);
for (int i = 1; i <= n; i++)
if (into[i] > outo[i]) addedge(s, i, (into[i]-outo[i])>>1);
else if (into[i] < outo[i]) addedge(i, t, (outo[i]-into[i])>>1);
return Dinic() == tot;
}
void bi_search(int l, int r) {
int ans = 0;
while (l <= r)
if (!chk(mid)) l = mid+1;
else ans = mid, r = mid-1;
if (!ans) puts("NIE");
else printf("%d\n", ans);
}
int main() {
read(n), read(m);
for (int i = 1, u, v, x, y; i <= m; i++) {
read(u), read(v), read(x), read(y);
if (x < y) e[i] = (edge){u, v, x, y};
else e[i] = (edge){v, u, y, x};
}
return bi_search(1, 1000), 0;
}
------------- Thanks For Reading -------------
0%